home *** CD-ROM | disk | FTP | other *** search
Text File | 1994-10-02 | 18.7 KB | 646 lines | [TEXT/ttxt] |
-
- #include <acc.h>
- #include <file.h>
- #include <os.h>
- #include <win.h>
- #include <qdvars.h>
- #include <event.h>
- #include <te.h>
- #include <dialog.h>
- #include <control.h>
- #include <device.h>
- #include <desk.h>
-
- ACC( 0x0400, 0, 0xfffb, 0, 9, "HFS Find " )
-
- #define MAXLEVEL 32 /* how deep we can go */
-
- DialogPtr dp; /* modeless dialog for string selection */
- DialogPtr rp; /* modal dialog for search results */
- cinfopbrec mycpb; /* for cat info */
- wdpbrec mywdpb; /* for vref of default volume */
-
- OSErr err;
- char name[MAXLEVEL][32]; /* names of path components */
-
- int modkeys; /* state of modifier keys at last event */
- int OurRef; /* our reference number */
- unsigned int resbase; /* our first resource */
-
- accopen( dctl, pb )
- dctlentry *dctl;
- paramblockrec *pb;
- {
- if ( dctl->dCtlWindow == 0 )
- {
- GrafPtr saveport;
-
- GetPort(&saveport);
- OurRef = dctl->dCtlRefNum;
- resbase = (unsigned)0xc000 | (-32 * ( 1 + OurRef ));
- dp = GetNewDialog( resbase, 0L, -1L );
- ((WindowPeek)dp)->windowKind = OurRef;
- rp = GetNewDialog( resbase+1, 0L, -1L );
- ((WindowPeek)rp)->windowKind = OurRef;
- dctl->dCtlWindow = (WindowPeek)dp;
- /* activate NAME box and select SEARCH WHOLE DISK */
- press(2); press(16);
- SetPort(saveport);
- }
- }
-
- accclose( dctl, pb )
- dctlentry *dctl;
- paramblockrec *pb;
- {
- DisposDialog( rp );
- DisposDialog( dp );
- dctl->dCtlWindow = 0;
- return 0;
- }
-
- accctl( dctl, pb )
- dctlentry *dctl;
- paramblockrec *pb;
- {
- EventRecord *erp;
- GrafPtr savedport;
- DialogPtr whatDialog;
- int whatItem, tf;
-
- GetPort( &savedport );
-
- switch ( pb->paramunion.CntrlParam.CSCode ) {
- EventRecord event;
- case accCursor: /* make null event so that insert can flash */
- event.what = 0;
- DialogSelect( &event, &whatDialog, &whatItem );
- break;
- case accEvent:
- /*
- * Megamax forgot to define the long we need in the paramunion.
- * Rather than add if myself, I just use diskBuff, which is
- * also a long.
- */
- erp = (EventRecord *)(pb->paramunion.CntrlParam.csParam.diskBuff);
- modkeys = erp->modifiers;
- ((WindowPeek)dp)->windowKind = dialogKind;
- if ( IsDialogEvent( erp ) ) {
- tf = DialogSelect( erp, &whatDialog, &whatItem );
- if ( tf && whatDialog == dp ) {
- SetPort( dp );
- press( whatItem );
- }
- }
- ((WindowPeek)dp)->windowKind = OurRef;
- break;
- case accCut:
- DlgCut( dp ); break;
- case accCopy:
- DlgCopy( dp ); break;
- case accPaste:
- DlgPaste( dp ); break;
- case accClear:
- dlgdelete( dp ); break;
- }
- SetPort( savedport );
- }
-
- accprime() {}
- accstatus() {}
-
-
- /*********************************************************************
- ***************** respond to activity in dialog *********************
- *********************************************************************/
- press( i )
- {
- int type;
- ControlHandle item;
- Rect r;
-
- if ( i == 1 ) { /* SEARCH button? */
- dosearch(); return;
- }
- GetDItem( dp, i, &type, &item, &r );
- if ( type == ChkCtrl + CtrlItem ) { /* check box ? */
- type = GetCtlValue( item );
- SetCtlValue( item, 1-type );
- return;
- }
- if ( 12 <= i && i <= 15 ) { /* type, creator, or a date ? */
- if ( modkeys & 0x100 ) { /* pretzel key pressed */
- fromfile( i );
- }
- return;
- }
- if ( i == 17 ) { /* CHDIR button ? */
- paramblkptr pb, askfile();
- if ( (pb = askfile()) == 0 )
- return;
- PBSetVol( pb, FALSE );
- return;
- }
- }
-
- fromfile( item )
- {
- paramblkptr pb, askfile();
- ControlHandle checkbox, editrecord;
- Rect r; int type;
- char buf[12], *cp;
- unsigned long ftime;
- char digit();
- datetimerec dtr;
-
- pb = askfile();
- if ( pb == 0 )
- return;
- switch ( item ) {
- case 14:
- cp = & pb->paramunion.FileParam.ioFlFndrInfo.fdCreator[0];
- goto ct;
- case 15:
- cp = & pb->paramunion.FileParam.ioFlFndrInfo.fdType[0];
- ct:
- buf[0] = *cp++; buf[1] = *cp++;
- buf[2] = *cp++; buf[3] = *cp++;
- buf[4] = 0; break;
- case 12:
- case 13:
- ftime = pb->paramunion.FileParam.ioFlMdDat;
- Secs2Date( ftime, &dtr );
- buf[0] = digit( dtr.month / 10 );
- buf[1] = digit( dtr.month );
- buf[2] = '/';
- buf[3] = digit( dtr.day / 10 );
- buf[4] = digit( dtr.day );
- buf[5] = '/';
- buf[6] = digit( dtr.year / 1000 );
- buf[7] = digit( dtr.year / 100 );
- buf[8] = digit( dtr.year / 10 );
- buf[9] = digit( dtr.year );
- buf[10] = 0;
- break;
- }
- GetDItem( dp, item, &type, &editrecord, &r );
- SetIText( editrecord, buf );
- SelIText( dp, item, 0, 0 );
-
- DrawDialog( dp );
- }
-
- char digit( num )
- int num;
- {
- return num % 10 + '0';
- }
-
- /*********************************************************************
- ****************************** search! ******************************
- *********************************************************************/
-
- int targets[3][50]; /* file name patterns to search for */
- int tN; /* true if there are some file names */
- long tType; /* type to search for */
- long tCreat; /* creator to search for */
- int tT, tC; /* true if search for type, creator */
- unsigned long tBefore, tAfter; /* before and after mod times */
- int tB, tA; /* true if search for before, after */
-
- char scratchText[40];
- dosearch() {
- ControlHandle checkbox, editrecord;
- Rect r; int type, i;
- unsigned long maketime();
-
- tN = 0;
- for ( i = 2; i <= 4; i++ ) {
- GetDItem( dp, i, &type, &checkbox, &r );
- if ( GetCtlValue( checkbox ) ) {
- GetDItem( dp, i+7, &type, &editrecord, &r );
- GetIText( editrecord, scratchText );
- compile( targets[i-2], scratchText );
- tN++;
- } else
- targets[i-2][0] = -10;
- }
- /*
- * get target type and creator
- */
- tT = tC = 0;
- GetDItem( dp, 8, &type, &checkbox, &r );
- tT = GetCtlValue( checkbox );
- if ( tT ) {
- GetDItem( dp, 15, &type, &editrecord, &r );
- GetIText( editrecord, scratchText );
- tType = *(long *)scratchText;
- }
- GetDItem( dp, 7, &type, &checkbox, &r );
- tC = GetCtlValue( checkbox );
- if ( tC ) {
- GetDItem( dp, 14, &type, &editrecord, &r );
- GetIText( editrecord, scratchText );
- tCreat = *(long *)scratchText;
- }
- /*
- * mod time before and after
- */
- GetDItem( dp, 5, &type, &checkbox, &r );
- tA = GetCtlValue( checkbox );
- if ( tA ) {
- GetDItem( dp, 12, &type, &editrecord, &r );
- GetIText( editrecord, scratchText );
- tAfter = maketime( scratchText );
- }
- GetDItem( dp, 6, &type, &checkbox, &r );
- tB = GetCtlValue( checkbox );
- if ( tB ) {
- GetDItem( dp, 13, &type, &editrecord, &r );
- GetIText( editrecord, scratchText );
- tBefore = maketime( scratchText );
- }
- /*
- * see if user want to search whole disk, and do search
- */
- GetDItem( dp, 16, &type, &checkbox, &r );
- SearchFiles( GetCtlValue( checkbox ) );
- }
-
- int AbortSearch;
- long eDirID[MAXLEVEL+1];
- int curdepth;
- char Volume[32];
- int toodeep;
-
- SearchFiles( fromroot )
- {
- AbortSearch = toodeep = 0;
- mywdpb.ioNamePtr = Volume;
- pbhgetvol( &mywdpb, FALSE );
- mycpb.ioCompletion = 0;
- mycpb.ioVRefNum = mywdpb.ioVRefNum;
- curdepth = 0;
- eDirID[0] = fromroot ? 2L : mywdpb.iowddirid;
- if ( !fromroot )
- Volume[0] = 0;
- EnumerateCat();
- HideWindow( rp ); SelectWindow( dp );
- }
-
- long eIndex[MAXLEVEL];
-
- EnumerateCat()
- {
- callSelf:
- CheckAbort();
- if ( curdepth >= MAXLEVEL ) {
- if ( toodeep == 0 ) {
- NoteAlert( resbase+2, 0L );
- toodeep = 1;
- }
- goto retHere;
- }
-
- if ( AbortSearch ) {
- if ( AbortSearch == 2 )
- AbortSearch = 0;
- if ( curdepth == 0 )
- return;
- else
- goto retHere;
- }
-
- eIndex[curdepth] = 1;
- do {
- mycpb.ioNamePtr = name[curdepth]; name[curdepth][0] = 0;
- mycpb.ioFDirIndex = eIndex[curdepth];
- mycpb.paramunion.dirinfo.iodrdirid = eDirID[curdepth];
-
- err = pbgetcatinfo(&mycpb,FALSE);
- #define FINF paramunion.hfileinfo.ioFlFndrInfo
- #define TYPE FINF.fdType
- #define CREAT FINF.fdCreator
- #define MTIME paramunion.hfileinfo.ioFlMdDat
- #define lType(x) (*(long *)&(x.TYPE))
- #define lCreat(x) (*(long *)&(x.CREAT))
- #define lTime(x) ((unsigned long )(x.MTIME))
- if ( err == 0 ) {
- if (
- (!tN || ( matches( targets[0], name[curdepth] )
- || matches( targets[1], name[curdepth] )
- || matches( targets[2], name[curdepth] )
- )
- )
- &&
- (!tT || tType == lType(mycpb))
- &&
- (!tC || tCreat == lCreat(mycpb))
- &&
- (!tA || tAfter <= lTime(mycpb))
- &&
- (!tB || tBefore >= lTime(mycpb))
- )
- dumppath( curdepth, (int)mycpb.ioFlAttrib,
- eDirID[curdepth] );
- if ( mycpb.ioFlAttrib & 0x10 ) {
- eDirID[++curdepth] =
- mycpb.paramunion.dirinfo.iodrdirid;
- goto callSelf;
- retHere:
- --curdepth;
- err = 0;
- }
- eIndex[curdepth]++;
- }
- } while ( err == 0 && !AbortSearch );
- if ( AbortSearch == 2 )
- AbortSearch = 0;
- if ( curdepth == 0 )
- return;
- else
- goto retHere;
- }
-
- char path[ 1200 ]; /* make big enough to hold a full path
- * spec for the longest path */
- dumppath( level, last, DirID )
- long DirID;
- {
- int i; char *pp, *sap();
-
- pp = path; *pp = 0;
- if ( Volume[0] ) {
- pp = sap( pp, Volume ); pp = sap( pp, ":" );
- } else
- if ( level || (last & 0x10) )
- pp = sap( pp, ":" );
- for ( i = 0; i <= level; i++ ) {
- pp = sap( pp, name[i] );
- pp = sap( pp, i == level ? "" : ":" );
- }
- sap( name[level], name[level] );
- ParamText( name[level], path, (last & 0x10) ? "Dir":"File", "" );
- ShowWindow( rp );
- SelectWindow( rp ); DrawDialog( rp );
- SetPort( rp );
-
- while ( 1 ) {
- ((WindowPeek)rp)->windowKind = dialogKind;
- ModalDialog( 0L, &i );
- ((WindowPeek)rp)->windowKind = OurRef;
- switch ( i ) {
- case 2:
- AbortSearch = 1; /* FALL THROUGH */
- case 1:
- return; break;
- case 3:
- *(long *)0x398 = DirID; break;
- case 6:
- AbortSearch = 2; return;
- }
- }
- }
-
- char * sap( str, new )
- register char * str, * new;
- {
- while ( *str++ = *new++ )
- ;
- return --str;
- }
-
- CheckAbort() {
- if ( Button() )
- AbortSearch = 1;
- }
-
- /*********************************************************************
- ***************** file name matching with wildcards *****************
- *********************************************************************/
-
- #define STAR -1 /* match zero or more of anything */
- #define QM -2 /* match one of anything */
- #define END -3 /* match end of name */
-
- #define QUOTE ':' /* turn off special next character */
- #define cSTAR '*' /* character for STAR */
- #define cQM '?' /* character for QM */
- #define cEND '$' /* character for END */
-
- compile( dest, src )
- register int * dest;
- register char * src;
- {
- register char cc;
-
- while ( 1 ) {
- switch ( cc = *src++ ) {
- case cSTAR:
- *dest++ = STAR; break;
- case cQM:
- *dest++ = QM; break;
- case cEND:
- *dest++ = (*src == '\0') ? END : cc; break;
- case '\0':
- *dest++ = cc; return; break;
- case QUOTE:
- if ( (*dest++ = *src++) == '\0' )
- return;
- break;
- default:
- if ( 'A' <= cc && cc <= 'Z' )
- cc += 'a' - 'A';
- *dest++ = cc;
- }
- }
- }
-
- matches( cp, np )
- register int * cp; /* the compiled pattern to match */
- register unsigned char * np; /* the file name to check out */
- {
- unsigned char * savenp;
- int * startcp;
-
- savenp = np;
- startcp = cp;
-
- while ( 1 ) {
- char nc;
- if ( *cp == STAR ) {
- savenp = np; /* so we can backtrack */
- while ( *cp == STAR )
- cp++;
- }
- if ( *cp == '\0' )
- return 1;
- if ( (nc = *np++) == '\0' )
- return (*cp == END) ? 1 : 0;
- if ( 'A' <= nc && nc <= 'Z' )
- nc += 'a' - 'A';
- if ( *cp++ == nc || cp[-1] == QM )
- continue;
- /*
- * failed to match, so we have to backtrack
- */
- np = savenp + 1;
- while ( *--cp != STAR )
- if ( cp == startcp )
- return 0;
- }
- }
-
- /*********************************************************************
- ******************** Interface to SFGetFile *************************
- *********************************************************************/
-
- #include <pack.h>
- #include <event.h>
-
- SFReply rep;
- Point corner;
-
- /*
- * whatfile gets a ref ( V or WD ) num and a name for a user selected file
- */
- whatfile( ref, name )
- int *ref; /* where to put reference number */
- char *name; /* where to put name */
- {
- boolean myfilter();
-
- SetPt( &corner, 70, 70 ); /* where to put the box/
- ShowCursor(); /* make sure us/
- rep.good = 1;
- SFGetFile( &corner, (char *)0, (int (*)())0, -1, "",
- (char *)0, &rep );
- if ( !rep.good )
- return -1;
- strcpy( name, rep.fName ); /* copy back name */
- *ref = rep.vRefNum; /* ...and refNum */
- return 0; /
- }
-
- /*
- * askfile returns a paramblkptr for a user selected file
- */
- paramblkptr askfile() {
- int ref, err;
- static paramblockrec pb;
-
- if ( whatfile( &ref, path ) )
- return (char *)0;
-
- pb.ioCompletion = 0;
- pb.ioNamePtr = path;
- pb.ioVRefNum = ref;
- pb.paramunion.FileParam.ioFVersNum = 0;
-
- err = PBGetFInfo( &pb, FALSE );
- if ( err )
- return (char *)0;
- return &pb;
- }
-
- /*********************************************************************
- *************************** date conversion *************************
- *********************************************************************/
-
- char * getnum( cp, where )
- register char *cp;
- register unsigned long *where;
- {
- for ( *where = 0; '0' <= *cp && *cp <= '9'; cp++ )
- *where = *where * 10 + *cp - '0';
- return cp;
- }
-
- /*
- * convert string to a time in seconds since Jan 1 1904.
- */
- unsigned long maketime( cp )
- register char *cp;
- {
- long ans;
- register char *tp;
-
- if ( *cp == '-' ) {
- unsigned long relative();
- return relative( ++cp );
- }
-
- for ( tp = cp; *tp; tp++ ) {
- unsigned long absdate();
- if ( *tp == '/' )
- return absdate( cp );
- }
-
- getnum( cp, &ans );
- return ans;
- }
-
- /*
- * Relative to now: nnD for nn days in past, nnH or nnM for
- * hours or minutes
- */
- unsigned long
- relative( cp )
- register char *cp;
- {
- long scale; long interval;
-
- cp = getnum( cp, &interval );
- switch ( *cp ) {
- default: case 'd': case 'D': scale = 86400; break;
- case 'h': case 'H': scale = 3600; break;
- case 'm': case 'M': scale = 60; break;
- }
- return *(unsigned long *)0x20c - scale * interval;
- }
-
- /*
- * date is MM/DD/YY or MM/DD/YYYY
- */
- unsigned long absdate( cp )
- register char *cp;
- {
- unsigned long month = 0, day = 0, year = 0;
- datetimerec dtr; unsigned long secs;
-
- cp = getnum( cp, &month );
- if ( *cp++ != '/' )
- return 0L;
- cp = getnum( cp, &day );
- if ( *cp++ != '/' )
- return 0L;
- cp = getnum( cp, &year );
- if ( month < 1 || month > 12 )
- return 0L;
- if ( year < 1900 )
- year += 1900;
- dtr.year = year;
- dtr.month = month;
- dtr.day = day;
- dtr.hour = dtr.minute = dtr.second = 0;
- Date2Secs( &dtr, &secs );
- return secs;
- }
-
- /*********************************************************************
- ********************** print debugging info *************************
- *********************************************************************/
-
- #ifdef APRINT
- #define STOPALERT 12346 /* alert for errors */
-
- aprintf( fmt, a1, a2, a3, a4, a5, a6 )
- char *fmt;
- {
- char buf[200];
- sprintf( buf, fmt, a1, a2, a3, a4, a5, a6 );
- ParamText( buf, "", "", "" );
- StopAlert( STOPALERT, 0L );
- }
- #endif
-